---
title: 评论功能
docs:
  - route: /docs/components/comment-node
    title: 评论叶子节点
  - route: /docs/components/comment-toolbar-button
    title: 评论工具栏按钮
  - route: /docs/components/block-discussion
    title: 区块讨论
---

<ComponentPreview name="discussion-demo" />

<PackageInfo>

## 功能特性

- **文本评论**: 以内联标注形式添加文本评论标记
- **重叠评论**: 支持同一段文本上的多个评论
- **草稿评论**: 在最终确定前创建评论草稿
- **状态追踪**: 跟踪评论状态和用户交互
- **讨论集成**: 与讨论插件配合实现完整协作功能

</PackageInfo>

## 套件使用

<Steps>

### 安装

最快捷的添加评论功能方式是使用 `CommentKit`，它包含预配置的 `commentPlugin` 和相关组件以及它们的 [Plate UI](/docs/installation/plate-ui) 组件。

<ComponentSource name="comment-kit" />

- [`CommentLeaf`](/docs/components/comment-node): 渲染评论文本标记
- [`BlockDiscussion`](/docs/components/block-discussion): 渲染集成评论功能的讨论界面

### 添加套件

```tsx
import { createPlateEditor } from 'platejs/react';
import { CommentKit } from '@/components/editor/plugins/comment-kit';

const editor = createPlateEditor({
  plugins: [
    // ...其他插件
    ...CommentKit,
  ],
});
```

</Steps>

## 手动配置

<Steps>

### 安装

```bash
npm install @platejs/comment
```

### 扩展评论插件

创建带有状态管理扩展配置的评论插件：

```tsx
import {
  type ExtendConfig,
  type Path,
  isSlateString,
} from 'platejs';
import {
  type BaseCommentConfig,
  BaseCommentPlugin,
  getDraftCommentKey,
} from '@platejs/comment';
import { toTPlatePlugin } from 'platejs/react';
import { CommentLeaf } from '@/components/ui/comment-node';

type CommentConfig = ExtendConfig<
  BaseCommentConfig,
  {
    activeId: string | null;
    commentingBlock: Path | null;
    hoverId: string | null;
    uniquePathMap: Map<string, Path>;
  }
>;

export const commentPlugin = toTPlatePlugin<CommentConfig>(
  BaseCommentPlugin,
  ({ editor }) => ({
    options: {
      activeId: null,
      commentingBlock: null,
      hoverId: null,
      uniquePathMap: new Map(),
    },
    render: {
      node: CommentLeaf,
    },
  })
);
```

- `options.activeId`: 当前激活评论ID，用于视觉高亮
- `options.commentingBlock`: 当前被评论区块的路径
- `options.hoverId`: 当前悬停评论ID，用于悬停效果
- `options.uniquePathMap`: 追踪评论解析唯一路径的映射表
- `render.node`: 指定 [`CommentLeaf`](/docs/components/comment-node) 来渲染评论文本标记

### 添加点击处理

添加点击处理来管理激活评论状态：

```tsx
export const commentPlugin = toTPlatePlugin<CommentConfig>(
  BaseCommentPlugin,
  ({ editor }) => ({
    handlers: {
      // 点击评论标记时设置激活评论
      onClick: ({ api, event, setOption, type }) => {
        let leaf = event.target as HTMLElement;
        let isSet = false;

        const unsetActiveComment = () => {
          setOption('activeId', null);
          isSet = true;
        };

        if (!isSlateString(leaf)) unsetActiveComment();

        while (leaf.parentElement) {
          if (leaf.classList.contains(`slate-${type}`)) {
            const commentsEntry = api.comment.node();

            if (!commentsEntry) {
              unsetActiveComment();
              break;
            }

            const id = api.comment.nodeId(commentsEntry[0]);
            setOption('activeId', id ?? null);
            isSet = true;
            break;
          }

          leaf = leaf.parentElement;
        }

        if (!isSet) unsetActiveComment();
      },
    },
    // ... 之前的options和render配置
  })
);
```

点击处理器追踪当前激活的评论：
- **检测评论点击**: 遍历DOM查找评论元素
- **设置激活状态**: 点击评论时更新`activeId`
- **清除状态**: 点击评论外部时取消`activeId`
- **视觉反馈**: 在评论组件中启用悬停/激活样式

### 扩展转换

扩展`setDraft`转换以增强功能：

```tsx
export const commentPlugin = toTPlatePlugin<CommentConfig>(
  BaseCommentPlugin,
  ({ editor }) => ({
    // ... 之前的配置
  })
)
  .extendTransforms(
    ({
      editor,
      setOption,
      tf: {
        comment: { setDraft },
      },
    }) => ({
      setDraft: () => {
        if (editor.api.isCollapsed()) {
          editor.tf.select(editor.api.block()![1]);
        }

        setDraft();

        editor.tf.collapse();
        setOption('activeId', getDraftCommentKey());
        setOption('commentingBlock', editor.selection!.focus.path.slice(0, 1));
      },
    })
  )
  .configure({
    node: { component: CommentLeaf },
    shortcuts: {
      setDraft: { keys: 'mod+shift+m' },
    },
  });
```

### 添加工具栏按钮

您可以在[工具栏](/docs/toolbar)中添加[`CommentToolbarButton`](/docs/components/comment-toolbar-button)来为选中文本添加评论。

### 添加插件

```tsx
import { createPlateEditor } from 'platejs/react';

const editor = createPlateEditor({
  plugins: [
    // ...其他插件
    commentPlugin,
  ],
});
```

### 讨论集成

评论插件可与[讨论插件](/docs/discussion)配合实现完整协作：

```tsx
import { discussionPlugin } from '@/components/editor/plugins/discussion-kit';

const editor = createPlateEditor({
  plugins: [
    // ...其他插件
    discussionPlugin,
    commentPlugin,
  ],
});
```

</Steps>

## 键盘快捷键

<KeyTable>
  <KeyTableItem hotkey="Cmd + Shift + M">
    在选中文本上添加评论
  </KeyTableItem>
</KeyTable>

## Plate Plus

<ComponentPreviewPro name="comment-pro" />

## 插件

### `CommentPlugin`

用于创建和管理文本评论的插件，具有状态追踪和讨论集成功能。

<API name="CommentPlugin">
<APIOptions>
  <APIItem name="activeId" type="string | null">
    当前激活评论ID，用于视觉高亮。内部用于状态追踪。
  </APIItem>
  <APIItem name="commentingBlock" type="Path | null">
    当前被评论区块的路径。
  </APIItem>
  <APIItem name="hoverId" type="string | null">
    当前悬停评论ID，用于悬停效果。
  </APIItem>
  <APIItem name="uniquePathMap" type="Map<string, Path>">
    追踪评论解析唯一路径的映射表。
  </APIItem>
</APIOptions>
</API>

## API

### `api.comment.has`

检查编辑器中是否存在指定ID的评论。

<API name="has">
<APIParameters>
  <APIItem name="options" type="{ id: string }">
    包含要检查评论ID的选项。
  </APIItem>
</APIParameters>
<APIReturns type="boolean">
  评论是否存在。
</APIReturns>
</API>

### `api.comment.node`

获取评论节点entry。

<API name="node">
<APIOptions type="EditorNodesOptions & { id?: string; isDraft?: boolean }" optional>
  查找节点的选项。
</APIOptions>
<APIReturns type="NodeEntry<TCommentText> | undefined">
  找到的评论节点entry。
</APIReturns>
</API>

### `api.comment.nodeId`

从叶子节点获取评论ID。

<API name="nodeId">
<APIParameters>
  <APIItem name="leaf" type="TCommentText">
    评论叶子节点。
  </APIItem>
</APIParameters>
<APIReturns type="string | undefined">
  找到的评论ID。
</APIReturns>
</API>

### `api.comment.nodes`

获取所有匹配选项的评论节点entry。

<API name="nodes">
<APIOptions type="EditorNodesOptions & { id?: string; isDraft?: boolean }" optional>
  查找节点的选项。
</APIOptions>
<APIReturns type="NodeEntry<TCommentText>[]">
  评论节点entry数组。
</APIReturns>
</API>

## 转换

### `tf.comment.removeMark`

从当前选区或指定位置移除评论标记。

<API name="removeMark" />

### `tf.comment.setDraft`

在当前选区设置草稿评论标记。

<API name="setDraft">
<APIOptions type="SetNodesOptions" optional>
  设置草稿评论的选项。
</APIOptions>
</API>

### `tf.comment.unsetMark`

从编辑器中取消设置指定ID的评论节点。

<API name="unsetMark">
<APIParameters>
  <APIItem name="options" type="{ id: string }">
    包含要取消评论ID的选项。
  </APIItem>
</APIParameters>
</API>

## 工具函数

### `getCommentCount`

获取评论节点中非草稿评论的数量。

<API name="getCommentCount">
<APIParameters>
  <APIItem name="node" type="TCommentText">
    评论节点。
  </APIItem>
</APIParameters>
<APIReturns type="number">
  评论数量。
</APIReturns>
</API>

### `getCommentKey`

基于提供的ID生成评论key。

<API name="getCommentKey">
<APIParameters>
  <APIItem name="id" type="string">
    评论ID。
  </APIItem>
</APIParameters>
<APIReturns type="string">
    生成的评论key。
</APIReturns>
</API>

### `getCommentKeyId`

从评论key中提取评论ID。

<API name="getCommentKeyId">
<APIParameters>
  <APIItem name="key" type="string">
    评论key。
  </APIItem>
</APIParameters>
<APIReturns type="string">
    提取的评论ID。
</APIReturns>
</API>

### `getCommentKeys`

返回给定节点中存在的评论key数组。

<API name="getCommentKeys">
<APIParameters>
  <APIItem name="node" type="TCommentText">
    要检查评论key的节点。
  </APIItem>
</APIParameters>
<APIReturns type="string[]">
    评论key数组。
</APIReturns>
</API>

### `getDraftCommentKey`

获取草稿评论使用的key。

<API name="getDraftCommentKey">
<APIReturns type="string">
    草稿评论key。
</APIReturns>
</API>

### `isCommentKey`

检查给定key是否为评论key。

<API name="isCommentKey">
<APIParameters>
  <APIItem name="key" type="string">
    要检查的key。
  </APIItem>
</APIParameters>
<APIReturns type="boolean">
    是否为评论key。
</APIReturns>
</API>

### `isCommentNodeById`

检查给定节点是否为指定ID的评论。

<API name="isCommentNodeById">
<APIParameters>
  <APIItem name="node" type="TNode">
    要检查的节点。
  </APIItem>
  <APIItem name="id" type="string">
    评论ID。
  </APIItem>
</APIParameters>
<APIReturns type="boolean">
    节点是否为指定ID的评论。
</APIReturns>
</API>

## 类型

### `TCommentText`

可包含评论的文本节点。

<API name="TCommentText">
<APIAttributes>
  <APIItem name="comment" type="boolean" optional>
    该文本节点是否包含评论。
  </APIItem>
  <APIItem name="comment_<id>" type="boolean" optional>
    按评论ID索引的评论数据。一个文本节点可包含多个评论。
  </APIItem>
</APIAttributes>
</API>